home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / ireclaim.c < prev    next >
C/C++ Source or Header  |  1996-10-27  |  6KB  |  201 lines

  1. /* Copyright (C) 1995, 1996 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* ireclaim.c */
  20. /* Interpreter's interface to garbage collector */
  21. #include "ghost.h"
  22. #include "errors.h"
  23. #include "gsstruct.h"
  24. #include "iastate.h"
  25. #include "dstack.h"            /* for dsbot, dsp, dict_set_top */
  26. #include "estack.h"            /* for esbot, esp */
  27. #include "ostack.h"            /* for osbot, osp */
  28. #include "opdef.h"            /* for defining init procedure */
  29. #include "store.h"            /* for make_array */
  30.  
  31. /* Import the static interpreter refs from interp.c. */
  32. extern ref ref_static_stacks;
  33. extern ref ref_ref_stacks[3];
  34.  
  35. /* Import preparation and cleanup routines. */
  36. extern void file_gc_prepare(P0());
  37. extern void dstack_gc_cleanup(P0());
  38.  
  39. /* Forward references */
  40. private void gs_vmreclaim(P2(gs_dual_memory_t *, bool));
  41.  
  42. /* Initialize the GC hook in the allocator. */
  43. private int ireclaim(P2(gs_dual_memory_t *, int));
  44. private void
  45. ireclaim_init(void)
  46. {    gs_imemory.reclaim = ireclaim;
  47. }
  48.  
  49. /* GC hook called when the allocator returns a VMerror (space = -1), */
  50. /* or for vmreclaim (space = the space to collect). */
  51. private int
  52. ireclaim(gs_dual_memory_t *dmem, int space)
  53. {    bool global;
  54.     gs_ref_memory_t *mem;
  55.     if ( space < 0 )
  56.       {    /* Determine which allocator got the VMerror. */
  57.         gs_memory_status_t stats;
  58.         int i;
  59.         mem = dmem->space_global;    /* just in case */
  60.         for ( i = 0; i < countof(dmem->spaces.indexed); ++i )
  61.           {    mem = dmem->spaces.indexed[i];
  62.             if ( mem == 0 )
  63.               continue;
  64.             if ( mem->gc_status.requested > 0 )
  65.               break;
  66.           }
  67.         gs_memory_status((gs_memory_t *)mem, &stats);
  68.         if ( stats.allocated >= mem->gc_status.max_vm )
  69.           {    /* We can't satisfy this request within max_vm. */
  70.             return_error(e_VMerror);
  71.           }
  72.       }
  73.     else
  74.       {    mem = dmem->spaces.indexed[space >> r_space_shift];
  75.       }
  76.     if_debug3('0', "[0]GC called, space=%d, requestor=%d, requested=%ld\n",
  77.           space, mem->space, (long)mem->gc_status.requested);
  78.     global = mem->space != avm_local;
  79.     gs_vmreclaim(dmem, global);
  80.     ialloc_set_limit(mem);
  81.     ialloc_reset_requested(dmem);
  82.     return 0;
  83. }
  84.  
  85. /* Interpreter entry to garbage collector. */
  86. /* This registers the stacks before calling the main GC. */
  87. private void near set_ref_chunk(P4(chunk_t *, ref *, ref *, gs_ref_memory_t *));
  88. private void
  89. gs_vmreclaim(gs_dual_memory_t *dmem, bool global)
  90. {    /*
  91.      * Create pseudo-chunks to hold the interpreter roots:
  92.      * copies of the ref_stacks, and, if necessary,
  93.      * the statically allocated stack bodies.
  94.      */
  95.     gs_ref_memory_t *lmem = dmem->space_local;
  96.     gs_ref_memory_t *gmem = dmem->space_global;
  97.     gs_ref_memory_t *smem = dmem->space_system;
  98.     struct ir_ {
  99.         chunk_head_t head;
  100.         obj_header_t prefix;
  101.         ref refs[5+1];        /* +1 for extra relocation ref */
  102.     } iroot_refs;
  103.     chunk_t cir, css;
  104.     void *piroot = &iroot_refs.refs[0];
  105.     gs_gc_root_t iroot;
  106.  
  107.     alloc_close_chunk(lmem);
  108.     if ( gmem != lmem )
  109.       alloc_close_chunk(gmem);
  110.     alloc_close_chunk(smem);
  111.  
  112.     /*
  113.      * Copy the ref_stacks into the heap, so we can trace and
  114.      * relocate them.  Note that they are allocated in system VM.
  115.      */
  116. #define stkmem smem
  117. #define get_stack(i, stk)\
  118.   ref_stack_cleanup(&stk);\
  119.   iroot_refs.refs[i+2] = ref_ref_stacks[i],\
  120.   *r_ptr(&iroot_refs.refs[i+2], ref_stack) = stk
  121.     get_stack(0, d_stack);
  122.     get_stack(1, e_stack);
  123.     get_stack(2, o_stack);
  124. #undef get_stack
  125.  
  126.     /* Make the root chunk. */
  127.     iroot_refs.refs[1] = ref_static_stacks;
  128.     make_array(&iroot_refs.refs[0], avm_system, 4, &iroot_refs.refs[1]);
  129.     set_ref_chunk(&cir, &iroot_refs.refs[0], &iroot_refs.refs[5], stkmem);
  130.     gs_register_ref_root((gs_memory_t *)stkmem, &iroot, &piroot,
  131.                  "gs_gc_main");
  132.  
  133.     /* If necessary, make the static stack chunk. */
  134. #define css_array iroot_refs.refs[1]
  135. #define css_base css_array.value.refs
  136.     if ( css_base != NULL )
  137.       set_ref_chunk(&css, css_base, css_base + r_size(&css_array), stkmem);
  138.  
  139.     /* Prune the file list so it won't retain potentially collectible */
  140.     /* files. */
  141.     file_gc_prepare();
  142.  
  143.     /* Do the actual collection. */
  144.     gs_reclaim(&dmem->spaces, global);
  145.  
  146.     /* Remove the temporary chunks. */
  147.     if ( css_base != NULL )
  148.       alloc_unlink_chunk(&css, stkmem);
  149.     gs_unregister_root((gs_memory_t *)stkmem, &iroot, "gs_gc_main");
  150.     alloc_unlink_chunk(&cir, stkmem);
  151. #undef css_array
  152. #undef css_base
  153.  
  154.     /* Update the static copies of the ref_stacks. */
  155. #define put_stack(i, stk)\
  156.   ref_ref_stacks[i].value.pstruct = iroot_refs.refs[i+2].value.pstruct,\
  157.   stk = *r_ptr(&iroot_refs.refs[i+2], ref_stack)
  158.     put_stack(0, d_stack);
  159.     put_stack(1, e_stack);
  160.     put_stack(2, o_stack);
  161. #undef put_stack
  162. #undef stkmem
  163.  
  164.     /* Update the cached value pointers in names. */
  165.  
  166.     dstack_gc_cleanup();
  167.  
  168.     /* Reopen the active chunks. */
  169.  
  170.     alloc_open_chunk(smem);
  171.     if ( gmem != lmem )
  172.       alloc_open_chunk(gmem);
  173.     alloc_open_chunk(lmem);
  174.  
  175.     /* Update caches */
  176.  
  177.     { uint dcount = ref_stack_count(&d_stack);
  178.       ref_systemdict = *ref_stack_index(&d_stack, dcount - 1);
  179.     }
  180.     dict_set_top();
  181. }
  182. private void near
  183. set_ref_chunk(chunk_t *cp, ref *bot, ref *top, gs_ref_memory_t *mem)
  184. {    obj_header_t *pre = (obj_header_t *)bot - 1;
  185.     chunk_head_t *head = (chunk_head_t *)pre - 1;
  186.     pre->o_large = 1;        /* not relocatable */
  187.     pre->o_lsize = 0;
  188.     pre->o_lmark = o_l_unmarked;
  189.     pre->o_size = (byte *)(top + 1) - (byte *)bot;
  190.     pre->o_type = &st_refs;
  191.     alloc_init_chunk(cp, (byte *)head, (byte *)(top + 1), false, NULL);    /* +1 for extra reloc ref */
  192.     cp->cbot = cp->ctop;
  193.     alloc_link_chunk(cp, mem);
  194.     make_int(top, 0);        /* relocation ref */
  195. }
  196.  
  197. /* ------ Initialization procedure ------ */
  198.  
  199. BEGIN_OP_DEFS(ireclaim_l2_op_defs) {
  200. END_OP_DEFS(ireclaim_init) }
  201.